/*
 *  This file is part of audacious-hotkey plugin for audacious
 *
 *  Copyright (c) 2007 - 2008  Sascha Hlusiak <contact@saschahlusiak.de>
 *  Name: gui.c
 *  Description: gui.c
 *
 *  Part of this code is from itouch-ctrl plugin.
 *  Authors of itouch-ctrl are listed below:
 *
 *  Copyright (c) 2006 - 2007 Vladimir Paskov <vlado.paskov@gmail.com>
 *
 *  Part of this code are from xmms-itouch plugin.
 *  Authors of xmms-itouch are listed below:
 *
 *  Copyright (C) 2000-2002 Ville Syrjälä <syrjala@sci.fi>
 *                         Bryn Davies <curious@ihug.com.au>
 *                         Jonathan A. Davis <davis@jdhouse.org>
 *                         Jeremy Tan <nsx@nsx.homeip.net>
 *
 *  audacious-hotkey is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  (at your option) any later version.
 *
 *  audacious-hotkey is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with audacious-hotkey; if not, write to the Free Software
 *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
 * USA.
 */

#include <libaudcore/i18n.h>
#include <libaudcore/preferences.h>

#include <libaudgui/gtk-compat.h>
#include <libaudgui/libaudgui-gtk.h>

#include <gdk/gdkkeysyms-compat.h>
#include <gdk/gdkx.h>
#include <gtk/gtk.h>

#include <X11/XKBlib.h>

#include "grab.h"
#include "gui.h"
#include "plugin.h"

typedef struct _KeyControls
{
    GtkWidget * keytext;
    GtkWidget * grid;
    GtkWidget * button;
    GtkWidget * combobox;

    HotkeyConfiguration hotkey;
    struct _KeyControls *next, *prev, *first;
} KeyControls;

static KeyControls * first_controls;

static void clear_keyboard(GtkWidget * widget, void * data);
static void add_callback(GtkWidget * widget, void * data);
static void destroy_callback();
static void ok_callback();

static const char * event_desc[] = {N_("Previous track"),
                                    N_("Play"),
                                    N_("Pause/Resume"),
                                    N_("Stop"),
                                    N_("Next track"),
                                    N_("Step forward"),
                                    N_("Step backward"),
                                    N_("Mute"),
                                    N_("Volume up"),
                                    N_("Volume down"),
                                    N_("Jump to file"),
                                    N_("Toggle player window(s)"),
                                    N_("Show On-Screen-Display"),
                                    N_("Toggle repeat"),
                                    N_("Toggle shuffle"),
                                    N_("Toggle stop after current"),
                                    N_("Raise player window(s)")};

static_assert(aud::n_elems(event_desc) == EVENT_MAX,
              "event_desc table is not up to date");

static void set_keytext(GtkWidget * entry, int key, int mask, int type)
{
    char * text = nullptr;

    if (key == 0 && mask == 0)
    {
        text = g_strdup(_("(none)"));
    }
    else
    {
        static const char * modifier_string[] = {
            "Control", "Shift", "Alt", "Mod2", "Mod3", "Super", "Mod5"};
        static const unsigned int modifiers[] = {
            ControlMask, ShiftMask, Mod1Mask, Mod2Mask,
            Mod3Mask,    Mod4Mask,  Mod5Mask};
        const char * strings[9];
        char * keytext = nullptr;
        int i, j;
        if (type == TYPE_KEY)
        {
            KeySym keysym;
            keysym = XkbKeycodeToKeysym(
                GDK_DISPLAY_XDISPLAY(gdk_display_get_default()), key, 0, 0);
            if (keysym == 0 || keysym == NoSymbol)
            {
                keytext = g_strdup_printf("#%d", key);
            }
            else
            {
                keytext = g_strdup(XKeysymToString(keysym));
            }
        }
        if (type == TYPE_MOUSE)
        {
            keytext = g_strdup_printf("Button%d", key);
        }

        for (i = 0, j = 0; j < 7; j++)
        {
            if (mask & modifiers[j])
                strings[i++] = modifier_string[j];
        }
        if (key != 0)
            strings[i++] = keytext;
        strings[i] = nullptr;

        text = g_strjoinv(" + ", (char **)strings);
        g_free(keytext);
    }

    gtk_entry_set_text(GTK_ENTRY(entry), text);
    gtk_editable_set_position(GTK_EDITABLE(entry), -1);
    if (text)
        g_free(text);
}

static gboolean on_entry_key_press_event(GtkWidget * widget,
                                         GdkEventKey * event, void * user_data)
{
    KeyControls * controls = (KeyControls *)user_data;
    int is_mod;
    int mod;

    if (event->keyval == GDK_Tab)
        return false;
    if (event->keyval == GDK_Escape && ((event->state & ~GDK_LOCK_MASK) == 0))
        return false;
    if (event->keyval == GDK_Return && ((event->state & ~GDK_LOCK_MASK) == 0))
        return false;
    if (event->keyval == GDK_ISO_Left_Tab)
    {
        set_keytext(controls->keytext, controls->hotkey.key,
                    controls->hotkey.mask, controls->hotkey.type);
        return false;
    }
    if (event->keyval == GDK_Up && ((event->state & ~GDK_LOCK_MASK) == 0))
        return false;
    if (event->keyval == GDK_Down && ((event->state & ~GDK_LOCK_MASK) == 0))
        return false;

    mod = 0;
    is_mod = 0;

    if ((event->state & GDK_CONTROL_MASK) |
        (!is_mod && (is_mod = (event->keyval == GDK_Control_L ||
                               event->keyval == GDK_Control_R))))
        mod |= ControlMask;

    if ((event->state & GDK_MOD1_MASK) |
        (!is_mod &&
         (is_mod = (event->keyval == GDK_Alt_L || event->keyval == GDK_Alt_R))))
        mod |= Mod1Mask;

    if ((event->state & GDK_SHIFT_MASK) |
        (!is_mod && (is_mod = (event->keyval == GDK_Shift_L ||
                               event->keyval == GDK_Shift_R))))
        mod |= ShiftMask;

    if ((event->state & GDK_MOD5_MASK) |
        (!is_mod && (is_mod = (event->keyval == GDK_ISO_Level3_Shift))))
        mod |= Mod5Mask;

    if ((event->state & GDK_MOD4_MASK) |
        (!is_mod && (is_mod = (event->keyval == GDK_Super_L ||
                               event->keyval == GDK_Super_R))))
        mod |= Mod4Mask;

    if (!is_mod)
    {
        controls->hotkey.key = event->hardware_keycode;
        controls->hotkey.mask = mod;
        controls->hotkey.type = TYPE_KEY;
        if (controls->next == nullptr)
            add_callback(nullptr, (void *)controls);
        else
            gtk_widget_grab_focus(GTK_WIDGET(controls->next->keytext));
    }

    set_keytext(controls->keytext, is_mod ? 0 : event->hardware_keycode, mod,
                TYPE_KEY);
    return true;
}

static gboolean on_entry_key_release_event(GtkWidget * widget,
                                           GdkEventKey * event,
                                           void * user_data)
{
    KeyControls * controls = (KeyControls *)user_data;
    if (!gtk_widget_is_focus(widget))
        return false;
    set_keytext(controls->keytext, controls->hotkey.key, controls->hotkey.mask,
                controls->hotkey.type);

    return true;
}

static gboolean on_entry_button_press_event(GtkWidget * widget,
                                            GdkEventButton * event,
                                            void * user_data)
{
    KeyControls * controls = (KeyControls *)user_data;
    int mod;

    if (!gtk_widget_is_focus(widget))
        return false;

    mod = 0;
    if (event->state & GDK_CONTROL_MASK)
        mod |= ControlMask;

    if (event->state & GDK_MOD1_MASK)
        mod |= Mod1Mask;

    if (event->state & GDK_SHIFT_MASK)
        mod |= ShiftMask;

    if (event->state & GDK_MOD5_MASK)
        mod |= Mod5Mask;

    if (event->state & GDK_MOD4_MASK)
        mod |= Mod4Mask;

    if ((event->button <= 3) && (mod == 0))
    {
        GtkWidget * dialog;
        int response;
        dialog = gtk_message_dialog_new(
            GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_MODAL,
            GTK_MESSAGE_WARNING, GTK_BUTTONS_YES_NO,
            _("It is not recommended to bind the primary mouse buttons without "
              "modifiers.\n\n"
              "Do you want to continue?"));
        gtk_window_set_title(GTK_WINDOW(dialog), _("Binding mouse buttons"));
        gtk_window_set_role(GTK_WINDOW(dialog), "message");
        response = gtk_dialog_run(GTK_DIALOG(dialog));
        gtk_widget_destroy(dialog);
        if (response != GTK_RESPONSE_YES)
            return true;
    }

    controls->hotkey.key = event->button;
    controls->hotkey.mask = mod;
    controls->hotkey.type = TYPE_MOUSE;
    set_keytext(controls->keytext, controls->hotkey.key, controls->hotkey.mask,
                controls->hotkey.type);
    if (controls->next == nullptr)
        add_callback(nullptr, (void *)controls);

    return true;
}

static gboolean on_entry_scroll_event(GtkWidget * widget,
                                      GdkEventScroll * event, void * user_data)
{
    KeyControls * controls = (KeyControls *)user_data;
    int mod;

    if (!gtk_widget_is_focus(widget))
        return false;

    mod = 0;
    if (event->state & GDK_CONTROL_MASK)
        mod |= ControlMask;

    if (event->state & GDK_MOD1_MASK)
        mod |= Mod1Mask;

    if (event->state & GDK_SHIFT_MASK)
        mod |= ShiftMask;

    if (event->state & GDK_MOD5_MASK)
        mod |= Mod5Mask;

    if (event->state & GDK_MOD4_MASK)
        mod |= Mod4Mask;

    if (event->direction == GDK_SCROLL_UP)
        controls->hotkey.key = 4;
    else if (event->direction == GDK_SCROLL_DOWN)
        controls->hotkey.key = 5;
    else if (event->direction == GDK_SCROLL_LEFT)
        controls->hotkey.key = 6;
    else if (event->direction == GDK_SCROLL_RIGHT)
        controls->hotkey.key = 7;
    else
        return false;

    controls->hotkey.mask = mod;
    controls->hotkey.type = TYPE_MOUSE;
    set_keytext(controls->keytext, controls->hotkey.key, controls->hotkey.mask,
                controls->hotkey.type);
    if (controls->next == nullptr)
        add_callback(nullptr, (void *)controls);
    return true;
}

KeyControls * add_event_controls(KeyControls * list, GtkWidget * grid, int row,
                                 HotkeyConfiguration * hotkey)
{
    KeyControls * controls;
    int i;

    controls = (KeyControls *)g_malloc(sizeof(KeyControls));
    controls->next = nullptr;
    controls->prev = list;
    controls->first = list->first;
    controls->grid = grid;
    list->next = controls;

    if (hotkey)
    {
        controls->hotkey.key = hotkey->key;
        controls->hotkey.mask = hotkey->mask;
        controls->hotkey.type = hotkey->type;
        controls->hotkey.event = hotkey->event;
        if (controls->hotkey.key == 0)
            controls->hotkey.mask = 0;
    }
    else
    {
        controls->hotkey.key = 0;
        controls->hotkey.mask = 0;
        controls->hotkey.type = TYPE_KEY;
        controls->hotkey.event = (EVENT)0;
    }

    controls->combobox = gtk_combo_box_text_new();
    for (i = 0; i < EVENT_MAX; i++)
    {
        gtk_combo_box_text_append_text((GtkComboBoxText *)controls->combobox,
                                       _(event_desc[i]));
    }
    gtk_combo_box_set_active(GTK_COMBO_BOX(controls->combobox),
                             controls->hotkey.event);

    controls->keytext = gtk_entry_new();
    gtk_editable_set_editable(GTK_EDITABLE(controls->keytext), false);

    set_keytext(controls->keytext, controls->hotkey.key, controls->hotkey.mask,
                controls->hotkey.type);
    g_signal_connect((void *)controls->keytext, "key_press_event",
                     G_CALLBACK(on_entry_key_press_event), controls);
    g_signal_connect((void *)controls->keytext, "key_release_event",
                     G_CALLBACK(on_entry_key_release_event), controls);
    g_signal_connect((void *)controls->keytext, "button_press_event",
                     G_CALLBACK(on_entry_button_press_event), controls);
    g_signal_connect((void *)controls->keytext, "scroll_event",
                     G_CALLBACK(on_entry_scroll_event), controls);

    controls->button = gtk_button_new();
    gtk_button_set_image(
        GTK_BUTTON(controls->button),
        gtk_image_new_from_icon_name("edit-delete", GTK_ICON_SIZE_BUTTON));
    g_signal_connect(G_OBJECT(controls->button), "clicked",
                     G_CALLBACK(clear_keyboard), controls);

#ifdef USE_GTK3
    gtk_widget_set_hexpand(controls->combobox, true);
    gtk_widget_set_hexpand(controls->keytext, true);

    gtk_grid_attach(GTK_GRID(grid), controls->combobox, 0, row, 1, 1);
    gtk_grid_attach(GTK_GRID(grid), controls->keytext, 1, row, 1, 1);
    gtk_grid_attach(GTK_GRID(grid), controls->button, 2, row, 1, 1);
#else
    gtk_table_attach_defaults(GTK_TABLE(grid), controls->combobox, 0, 1, row,
                              row + 1);
    gtk_table_attach_defaults(GTK_TABLE(grid), controls->keytext, 1, 2, row,
                              row + 1);
    gtk_table_attach_defaults(GTK_TABLE(grid), controls->button, 2, 3, row,
                              row + 1);
#endif

    gtk_widget_grab_focus(GTK_WIDGET(controls->keytext));
    return controls;
}

void * make_config_widget()
{
    KeyControls * current_controls;
    GtkWidget * main_vbox, * hbox;
    GtkWidget * frame;
    GtkWidget * label1, * label2, * label3;
    GtkWidget * image;
    GtkWidget * grid;
    GtkWidget * button_box, * button;
    PluginConfig * plugin_cfg;
    HotkeyConfiguration * hotkey, temphotkey;
    int i;

    load_config();

    plugin_cfg = get_config();

    ungrab_keys();

    main_vbox = audgui_vbox_new(4);

    hbox = audgui_hbox_new(2);
    gtk_box_pack_start(GTK_BOX(main_vbox), hbox, false, true, 0);
    image = gtk_image_new_from_icon_name("dialog-information",
                                         GTK_ICON_SIZE_DIALOG);
    gtk_box_pack_start(GTK_BOX(hbox), image, false, true, 0);
    label1 = gtk_label_new(_("Press a key combination inside a text field.\nYou "
                             "can also bind mouse buttons."));
#ifdef USE_GTK3
    gtk_widget_set_halign(label1, GTK_ALIGN_START);
#else
    gtk_misc_set_alignment(GTK_MISC(label1), 0, 0.5);
#endif
    gtk_box_pack_start(GTK_BOX(hbox), label1, true, true, 0);

    label1 = gtk_label_new(nullptr);
    gtk_label_set_markup(GTK_LABEL(label1), _("Hotkeys:"));
    frame = gtk_frame_new(nullptr);
    gtk_frame_set_label_widget(GTK_FRAME(frame), label1);
    gtk_box_pack_start(GTK_BOX(main_vbox), frame, true, true, 0);
    gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_ETCHED_IN);

    grid = audgui_grid_new();
    audgui_grid_set_column_spacing(grid, 2);
    gtk_container_set_border_width(GTK_CONTAINER(grid), 3);
    gtk_container_add(GTK_CONTAINER(frame), grid);

    label2 = gtk_label_new(nullptr);
    gtk_label_set_justify(GTK_LABEL(label2), GTK_JUSTIFY_CENTER);
    gtk_label_set_markup(GTK_LABEL(label2), _("<b>Action:</b>"));

    label3 = gtk_label_new(nullptr);
    gtk_label_set_justify(GTK_LABEL(label3), GTK_JUSTIFY_CENTER);
    gtk_label_set_markup(GTK_LABEL(label3), _("<b>Key Binding:</b>"));

#ifdef USE_GTK3
    gtk_grid_attach(GTK_GRID(grid), label2, 0, 0, 1, 1);
    gtk_grid_attach(GTK_GRID(grid), label3, 1, 0, 1, 1);
#else
    gtk_misc_set_alignment(GTK_MISC(label2), 0.5, 0.5);
    gtk_misc_set_alignment(GTK_MISC(label3), 0.5, 0.5);

    gtk_table_attach_defaults(GTK_TABLE(grid), label2, 0, 1, 0, 1);
    gtk_table_attach_defaults(GTK_TABLE(grid), label3, 1, 2, 0, 1);
#endif

    hotkey = &(plugin_cfg->first);
    i = 1;
    first_controls = (KeyControls *)g_malloc(sizeof(KeyControls));
    first_controls->next = nullptr;
    first_controls->prev = nullptr;
    first_controls->grid = grid;
    first_controls->button = nullptr;
    first_controls->combobox = nullptr;
    first_controls->keytext = nullptr;
    first_controls->first = first_controls;
    first_controls->hotkey.key = 0;
    first_controls->hotkey.mask = 0;
    first_controls->hotkey.event = (EVENT)0;
    first_controls->hotkey.type = TYPE_KEY;
    current_controls = first_controls;
    if (hotkey->key != 0)
    {
        while (hotkey)
        {
            current_controls =
                add_event_controls(current_controls, grid, i, hotkey);
            hotkey = hotkey->next;
            i++;
        }
    }
    temphotkey.key = 0;
    temphotkey.mask = 0;
    temphotkey.type = TYPE_KEY;
    if (current_controls != first_controls)
        temphotkey.event = (EVENT)(current_controls->hotkey.event + 1);
    else
        temphotkey.event = (EVENT)0;
    if (temphotkey.event >= EVENT_MAX)
        temphotkey.event = (EVENT)0;
    add_event_controls(current_controls, grid, i, &temphotkey);

    hbox = audgui_hbox_new(0);
    gtk_box_pack_start(GTK_BOX(main_vbox), hbox, false, true, 0);

    button_box = audgui_button_box_new(GTK_ORIENTATION_HORIZONTAL);
    gtk_box_pack_start(GTK_BOX(hbox), button_box, false, true, 0);
    gtk_button_box_set_layout(GTK_BUTTON_BOX(button_box), GTK_BUTTONBOX_START);
    gtk_box_set_spacing(GTK_BOX(button_box), 4);

    button = audgui_button_new(_("_Add"), "list-add", nullptr, nullptr);
    gtk_container_add(GTK_CONTAINER(button_box), button);
    g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(add_callback),
                     first_controls);

    return main_vbox;
}

static void clear_keyboard(GtkWidget * widget, void * data)
{
    KeyControls * controls = (KeyControls *)data;

    if ((controls->next == nullptr) && (controls->prev->keytext == nullptr))
    {
        controls->hotkey.key = 0;
        controls->hotkey.mask = 0;
        controls->hotkey.type = TYPE_KEY;
        set_keytext(controls->keytext, 0, 0, TYPE_KEY);
        gtk_combo_box_set_active(GTK_COMBO_BOX(controls->combobox), 0);
        return;
    }

    if (controls->prev)
    {
        KeyControls * c;
        GtkWidget * grid;
        int row;

        gtk_widget_destroy(GTK_WIDGET(controls->button));
        gtk_widget_destroy(GTK_WIDGET(controls->keytext));
        gtk_widget_destroy(GTK_WIDGET(controls->combobox));

        row = 0;
        c = controls->first;
        while (c)
        {
            if (c == controls)
                break;
            row++;
            c = c->next;
        }
        c = controls->next;
        controls->prev->next = controls->next;
        if (controls->next)
            controls->next->prev = controls->prev;
        g_free(controls);
        if (c)
            grid = c->grid;
        else
            grid = nullptr;
        while (c)
        {
            g_object_ref(c->combobox);
            g_object_ref(c->keytext);
            g_object_ref(c->button);

            gtk_container_remove(GTK_CONTAINER(c->grid), c->combobox);
            gtk_container_remove(GTK_CONTAINER(c->grid), c->keytext);
            gtk_container_remove(GTK_CONTAINER(c->grid), c->button);

#ifdef USE_GTK3
            gtk_grid_attach(GTK_GRID(c->grid), c->combobox, 0, row, 1, 1);
            gtk_grid_attach(GTK_GRID(c->grid), c->keytext, 1, row, 1, 1);
            gtk_grid_attach(GTK_GRID(c->grid), c->button, 2, row, 1, 1);
#else
            gtk_table_attach_defaults(GTK_TABLE(c->grid), c->combobox, 0, 1,
                                      row, row + 1);
            gtk_table_attach_defaults(GTK_TABLE(c->grid), c->keytext, 1, 2, row,
                                      row + 1);
            gtk_table_attach_defaults(GTK_TABLE(c->grid), c->button, 2, 3, row,
                                      row + 1);
#endif
            g_object_unref(c->combobox);
            g_object_unref(c->keytext);
            g_object_unref(c->button);

            c = c->next;
            row++;
        }
        if (grid)
            gtk_widget_show_all(GTK_WIDGET(grid));

        return;
    }
}

void add_callback(GtkWidget * widget, void * data)
{
    KeyControls * controls = (KeyControls *)data;
    HotkeyConfiguration temphotkey;
    int count;
    if (controls == nullptr)
        return;
    if ((controls->next == nullptr) &&
        (controls->hotkey.event + 1 == EVENT_MAX))
        return;
    controls = controls->first;
    if (controls == nullptr)
        return;
    count = 1;
    while (controls->next)
    {
        controls = controls->next;
        count = count + 1;
    }
    temphotkey.key = 0;
    temphotkey.mask = 0;
    temphotkey.type = TYPE_KEY;
    temphotkey.event = (EVENT)(controls->hotkey.event + 1);
    if (temphotkey.event >= EVENT_MAX)
        temphotkey.event = (EVENT)0;
    add_event_controls(controls, controls->grid, count, &temphotkey);
    gtk_widget_show_all(GTK_WIDGET(controls->grid));
}

void destroy_callback()
{
    KeyControls * controls = first_controls;

    grab_keys();

    while (controls)
    {
        KeyControls * old;
        old = controls;
        controls = controls->next;
        g_free(old);
    }

    first_controls = nullptr;
}

void ok_callback()
{
    KeyControls * controls = first_controls;
    PluginConfig * plugin_cfg = get_config();
    HotkeyConfiguration * hotkey;

    hotkey = &(plugin_cfg->first);
    hotkey = hotkey->next;
    while (hotkey)
    {
        HotkeyConfiguration * old;
        old = hotkey;
        hotkey = hotkey->next;
        g_free(old);
    }
    plugin_cfg->first.next = nullptr;
    plugin_cfg->first.key = 0;
    plugin_cfg->first.event = (EVENT)0;
    plugin_cfg->first.mask = 0;

    hotkey = &(plugin_cfg->first);
    while (controls)
    {
        if (controls->hotkey.key)
        {
            if (hotkey->key)
            {
                hotkey->next = g_new(HotkeyConfiguration, 1);
                hotkey = hotkey->next;
                hotkey->next = nullptr;
            }
            hotkey->key = controls->hotkey.key;
            hotkey->mask = controls->hotkey.mask;
            hotkey->event = (EVENT)gtk_combo_box_get_active(
                GTK_COMBO_BOX(controls->combobox));
            hotkey->type = controls->hotkey.type;
        }
        controls = controls->next;
    }

    save_config();
}

static const PreferencesWidget hotkey_widgets[] = {
    WidgetCustomGTK(make_config_widget)};

const PluginPreferences hotkey_prefs = {{hotkey_widgets},
                                        nullptr, // init
                                        ok_callback,
                                        destroy_callback};
